You Should Know JavaScript
JS Map and Set
Map
- Map is a collection of keyed data items, just like an Object. But the main difference is that Map allows keys of any type.
- Avoid using
map[key]=value
ormap[key]
to set and get value from a map, instead use set and get methods
new Map()
– creates the map.
map.set(key, value)
– stores the value by the key.
map.get(key)
– returns the value by the key, undefined if key doesn’t exist in map.
map.has(key)
– returns true if the key exists, false otherwise.
map.delete(key)
– removes the element (the key/value pair) by the key.
map.clear()
– removes everything from the map.
map.size
– returns the current element count.
let map = new Map();
map.set('1', 'str1'); // a string key
map.set(1, 'num1'); // a numeric key
map.set(true, 'bool1'); // a boolean key
// remember the regular Object? it would convert keys to string
// Map keeps the type, so these two are different:
console.log(map.get(1)); // 'num1'
console.log(map.get('1') ); // 'str1'
console.log(map.size); // 3
Map Vs Object
- For js object key must be string, otherwise it will be converted to string, however in map key can be anything(object, primitives)
const obj = {};
obj[12] = 'number of columns';
obj[{name: 'John'}] = 450;
console.log(obj); //{ '12': 'number of columns', '[object Object]': 450 }
const m = new Map();
m.set(12, 'number of columns');
m.set({name: 'John'}, 450);
console.log(m); //Map(2) { 12 => 'number of columns', { name: 'John' } => 450 }
Iteration
const priceMap = new Map([
['mac', 2400],
['iphone', 1200],
['samsung', 900]
]); //map can be created with array
for (let brand of priceMap.keys()) {
console.log(brand, priceMap.get(brand));
}
//or
priceMap.forEach((value,key, map) => {
console.log(key, value);
})
Set
- A Set is a special type collection – “set of values” (without keys), where each value may occur only once.
- Like map key can be any js type(number, boolean, array, object…)
new Set([iterable])
– creates the set, and if an iterable object is provided (usually an array), copies values from it into the set.
set.add(value)
– adds a value, returns the set itself.
set.delete(value)
– removes the value, returns true if value existed at the moment of the call, otherwise false.
set.has(value)
– returns true if the value exists in the set, otherwise false.
set.clear()
– removes everything from the set.
set.size
– is the elements count.
let set = new Set();
let john = { name: "John" };
let pete = { name: "Pete" };
let mary = { name: "Mary" };
// visits, some users come multiple times
set.add(john);
set.add(pete);
//set.add([1,2,34]);
set.add(john);
set.add(mary);
// set keeps only unique values
console.log( set.size ); // 3
for (let user of set) {
console.log(user); // John (then Pete and Mary)
}
Iteration
let set = new Set(["oranges", "apples", "bananas"]);
for (let value of set) console.log(value);
// the same with forEach:
set.forEach((value, valueAgain, set) => {
console.log(value);
});
Higher Order Functions
A function which takes another function as an argument or returns a function is known as a higher order function.
Filter
- Returns a new array with any elements for which the callback function returns
true
.
const words = ['spray', 'limit', 'elite', 'exuberant', 'destruction', 'present'];
const result = words.filter(word => word.length > 6);
console.log(result);
Map
- Creates a new array populated with the results of calling a provided function on every element in the calling array.
- The original array does not get altered
const array1 = [1, 4, 9, 16];
// pass a function to map
const map1 = array1.map(x => x * 2);
console.log(map1);
// expected output: Array [2, 8, 18, 32]
Foreach
- Executes a callback function on each of the elements in an array in order.
const numbers = [28, 77, 45, 99, 27];
numbers.forEach(number => {
console.log(number);
});
Reduce
- Iterates through an array and returns a single value
const arrayOfNumbers = [1, 2, 3, 4];
const sum = arrayOfNumbers.reduce((accumulator, currentValue) => {
return accumulator + currentValue;
});
- We can pass initial value
const array1 = [1, 2, 3, 4];
// 0 + 1 + 2 + 3 + 4
const initialValue = 0;
const sumWithInitial = array1.reduce(
(accumulator, currentValue) => accumulator + currentValue,
initialValue
);
console.log(sumWithInitial);
// expected output: 10
Find
- Returns the first element in the provided array that satisfies the provided testing function.
- If no values satisfy the testing function,
undefined
is returned.
const array1 = [5, 12, 8, 130, 44];
const found = array1.find(element => element > 10);
console.log(found);
// expected output: 12
FindLast
- Returns the value of the last element in an array that satisfies the provided testing function.
- If no elements satisfy the testing function,
undefined
is returned.
const array1 = [5, 12, 50, 130, 44];
const found = array1.findLast((element) => element > 45);
console.log(found);
// expected output: 130
Every
- The
every()
method tests whether all elements in the array pass the test implemented by the provided function. - It returns a Boolean value.
const array1 = [1, 30, 39, 29, 10, 13];
console.log(array1.every(val => val < 40));
// expected output: true
Some
- The
some()
method tests whether at least one element in the array passes the test implemented by the provided function. - It returns a Boolean value.
const array = [1, 2, 3, 4, 5];
// checks whether an element is even
const even = (element) => element % 2 === 0;
console.log(array.some(even));
// expected output: true
FindIndex
- Returns the index of the first element in an array that satisfies the provided testing function.
- If no elements satisfy the testing function, -1 is returned.
const array1 = [5, 12, 8, 130, 44];
const isLargeNumber = (element) => element > 13;
console.log(array1.findIndex(isLargeNumber));
// expected output: 3
FindLastIndex
- Returns the index of the last element in an array that satisfies the provided testing function.
- If no elements satisfy the testing function, -1 is returned.
const array1 = [5, 12, 50, 130, 44];
const isLargeNumber = (element) => element > 45;
console.log(array1.findLastIndex(isLargeNumber));
// expected output: 3 (of element with value: 30)
Data Manipulation Snippets for JS/React
let data = {
todos: []
};
//const [state, setState] = useState(data);
//add
data = { todos: [...data.todos, newData] }
//setState(data)
//remove by id
data = { todos: data.todos.filter(todo => todo.id !== id) }
//update by id
data = { todos: data.todos.map(todo => {
if(todo.id === id){
todo.status = 'updated';
}
return todo;
}) };
//add new entry
data = {...data, notes: []}
Named Export Vs Default Export in ES6/React
Named Export: (export)
- Can have multiple named exports per file.
- You import the specific exports you want surrounded in braces
- The name of imported module has to be the same as the name of the exported module. Or you need to use
as
// exports from ./MyComponent.js file
export const MyComponent = () => {}
export const MyComponent2 = () => {}
// imports
// ex. importing a single named export
import { MyComponent } from "./MyComponent";// ex. importing multiple named exports
import { MyComponent, MyComponent2 } from "./MyComponent";
import * as MainComponents from "./MyComponent";
// use MainComponents.MyComponent and MainComponents.MyComponent2 here
// ex. giving a named import a different name by using "as":
import { MyComponent2 as MyNewComponent } from "./MyComponent";
Named exports are useful to export several values. During the import, one will be able to use the same name to refer to the corresponding value.
Default Export: (export default)
- One can have only one default export per file.
- When we import we have to specify a name.
const MyComponent = () => {}
export default MyComponent;
// import
import MyDefaultComponent from "./MyDefaultExport";
Web Worker
- Mozilla documentation
- Web Workers are a simple means for web content to run scripts in background threads. The worker thread can perform tasks without interfering with the user interface.
- What you can and can't inside web workers
- You can run whatever code you like inside the worker thread, with some exceptions.
- For example, you can't directly manipulate the
DOM
from inside a worker, or use some default methods and properties of the window object
- For example, you can't directly manipulate the
- You can use a large number of items available under window, including
WebSockets
, and data storage mechanisms likeIndexedDB
.
- You can run whatever code you like inside the worker thread, with some exceptions.
Create a Web Worker
const worker = new Worker('worker-file.js');
Send Message & Responses
- Both the client and server use
postMessage
to send messages & responses to each other- The client uses it to send request
- The server uses it to send response
const worker = new Worker('worker-file.js');
myWorker.postMessage([first.value, second.value]);
Receive Message & Response
- Both client and server respond to messages via the
onmessage
event handler- The message is contained within the message event's data attribute.
- The data is copied rather than shared.
//webworker.js
//receive message from server side
onmessage = function(e) {
console.log('Worker: Message received from main script');
const result = e.data[0] * e.data[1];
if (isNaN(result)) {
postMessage('Please write two numbers'); //send the message to client
} else {
const workerResult = 'Result: ' + result;
console.log('Worker: Posting message back to main script');
postMessage(workerResult); //send the message to client
}
}
const worker = new Worker('worker-file.js');
myWorker.postMessage([first.value, second.value]);
//client can receive response with onmessage
myWorker.onmessage = function(e) {
result.textContent = e.data;
console.log('Message received from worker');
}
JS Sorting
Sorting is in-place in JavaScript
- Sort linear array
const array = [5,4,3,2,1];
array.sort()
console.log(array)
- Sort objects by
- String
- Numeric value
const array = [
{
id: 4,
name: "Zeku"
},
{
id: 1,
name: "Abel"
}
]
//sort by string value
array.sort((a,b) => a.name.localeCompare(b.name));
console.log(array)
const array = [
{
id: 4,
name: "Zeku"
},
{
id: 1,
name: "Abel"
}
]
//sort by numberic value
array.sort((a,b) => a.id - b.id);
console.log(array)
Slice Vs Splice
Slice
Syntax => slice(start, end), end is not included
- Return portion of an array
const animals = ['ant', 'bison', 'camel', 'duck', 'elephant'];
console.log(animals.slice(0, 2));
Splice
Syntax => splice(start_index, number_of_elts_to_remove, …list_of_elts_to_add)
- Remove elets with Splice
let array = [1,2,3,4,5];
array.splice(3, 1); //removes number 4
console.log(array)
- Add elements with slice
let array = [1,2,3,4,5];
array.splice(3, 0, 'A', 'B', 'C'); //Add A,B,C
console.log(array)
What is the Difference Between Var, Let, Const
- There are 3 type of scopes
- global
- local
- lexical scope or block scope
Variables declared with var are either function-scoped or global-scoped, depending on whether they are declared within a function or outside a function. variables declared with either const or let are block scope
- let & const
- block scope
- var
- function scope or global scope
- const
- can not be re-initialized
What is Closure In JavaScript?
- Closures - JavaScript | MDN
- A closure gives you access to an outer function's scope from an inner function.
function makeFunc() {
var name = "Mozilla"; // name is a local variable created by init
return function displayName() {
// displayName() is the inner function, that forms the closure
console.log(name); // use variable declared in the parent function
}
}
const res = makeFunc();
res();
- Code explanation
- In some programming languages, the local variables within a function exist for just the duration of that function's execution.
- Once makeFunc() finishes executing, you might expect that the name variable would no longer be accessible. However, because the code still works as expected, this is obviously not the case in JavaScript.
function add(x) {
return function (y) {
return x + y;
};
}
const res = add(14)(14) //res = 14
- Real world example of using closures